from datetime import datetime
import random
import re

from flask import current_app
from flask import request, make_response, jsonify
from flask import session

from info import constants, db
from info.libs.yuntongxun.sms import CCP
from info.models import User
from info.utils.response_code import RET

from . import passport_blue
from info.utils.captcha.captcha import captcha
from info import redis_store

"""用户退出"""
@passport_blue.route("/logout",methods=["GET","POST"])
def logout():
    # 退出的本质就是把把存在服务器里面session删除，然后再重新刷新页面就可以了
    session.pop("user_id",None)
    session.pop("nick_name",None)
    session.pop("mobile",None)
    session.pop("is_admin",None)

    return jsonify(erron=RET.OK,errmsg="退出成功")

"""用户登陆"""
@passport_blue.route("/login",methods = ["GET","POST"])
def login():
    mobile = request.json.get("mobile")
    password = request.json.get("password")
    # 通过手机号查询当前是否有这个用户
    try:
        user = User.query.filter(User.mobile == mobile).first()
    except Exception as e:
        # 把错误信息存储到log日志里面
        current_app.logger.error(e)
    # 判断是否有这个用户
    if not user:
        return jsonify(errno = RET.NODATA,errmsg = "请注册")

    # 通过系统的源码帮我们检查用户的密码是否正确
    if not user.check_password(password):
        return jsonify(errno = RET.PWDERR,errmsg = "请输入正确的密码")

    # 对用户进行状态保持,跟网易新闻一样,session进行实现,保持用户信息到session里面
    session["user_id"] = user.id
    session["nick_name"] = user.nick_name
    session["mobile"] = user.mobile
    # 更新最后登陆的时间
    user.last_login = datetime.now()
    # 把数据提交到数据库
    db.session.commit()

    return jsonify(errno = RET.OK,errmsg = "登陆成功")

# @passport_blue.route("/login",methods=["GET","POST"])
# def login():
#     # 从前端获取手机号和密码
#     mobile=request.json.get("mobile")
#     password=request.json.get("password")
#
#     # 从数据库中查找是否有这个手机号，并且进行查找错误拦截
#     try:
#         user=User.query.filter(User.mobile==mobile).first()
#     except Exception as e:
#         # 要是有错误就把错误日志存到日志中
#         current_app.logger.error(e)
#     # 如果查不到用户名
#     if not user:
#         return jsonify(erron=RET.NODATA,errmsg="请您先注册完整信息")
#     # 通过系统的源码帮助我们判断用户密码是否正确，源码在保存时，先把数据加密，校验时，需要解密再进行校验
#     if not user.check_password(password):
#         return jsonify(erron=RET.PWDERR,errmsg="请输入正确的密码")
#     # 走到这里说明用户的用户名和密码都正确，就可以把它存到session中，进行登陆保存（比方说保存一个月）
#     session["user_id"]=user.id
#     session["mobile"]=user.mobile
#     session["nick_name"]=user.nick_name
#     # 记录登陆的时间
#     user.last_agin=datetime.now()
#     # 提交到数据库
#     db.session.commit()
#
#     return jsonify(erron=RET.OK,errmsg="登陆成功")

"""用户注册"""

@passport_blue.route("/register",methods = ["GET","POST"])
def register():
    mobile = request.json.get("mobile")
    # 用户在客户端输入的短信验证码
    smscode = request.json.get("smscode")
    # 用户输入的密码
    password = request.json.get("password")
    # 获取到redis服务器里面存储的短信验证码
    real_sms_code = redis_store.get("sms_code_"+mobile)

    if not real_sms_code:
        return jsonify(errno = RET.NODATA,errmsg = "短信验证码已经过期")
    # 判断用户输入的短信验证码和服务器里面的短信验证码是否一致
    if smscode != real_sms_code:
        return jsonify(errno = RET.PARAMERR,errmsg = "请输入正确的短信验证码")

    # 创建一个用户对象用来注册用户
    user = User()
    user.mobile = mobile
    user.password = password
    user.nick_name = mobile
    # 获取到当前的时间,用来注册
    user.last_login = datetime.now()
    # 往数据库进行持久化操作
    db.session.add(user)
    db.session.commit()
    return jsonify(errno = RET.OK,errmsg = "注册成功")

# @passport_blue.route("/register",methods=["GET","POST"])
# def register():
#     mobile=request.json.get("mobile")
#     smscode=request.json.get("smscode")
#     password=request.json.get("password")
#     # 从redis里面获取短信验证码
#     real_sms_code=redis_store.get("sms_code_"+mobile)
#     # 判断验证码是否已过期
#     if not real_sms_code:
#         return jsonify(erron=RET.NODATA,errmsg="您的短信验证码已过期")
#     # 判断客户前段输入的验证码和从redis里面的验证码是否一致
#     if smscode != real_sms_code:
#         return jsonify(erron=RET.PARAMERR,errmsg="请输入正确的短信验证码")
#
#     # 程序走到这里说明用户的验证码是正确的，那么把客户的注册信息存储到mysql数据库中，存储的数据有用户手机号，密码
#     # 初始化User表对象
#     user= User()
#     user.mobile=mobile
#     user.password=password
#     user.nick_name=mobile
#     user.last_login=datetime.now()
#
#     db.session.add(user)
#     db.session.commit()
#     return jsonify(erron=RET.OK,errmsg="恭喜您，注册成功！！！")
"""
短信验证码
发送短信验证码实现流程：
    接收前端发送过来的请求参数
    检查参数是否已经全部传过来
    判断手机号格式是否正确
    检查图片验证码是否正确，若不正确，则返回

    生成随机的短信验证码
    使用第三方SDK发送短信验证码
"""
@passport_blue.route("/sms_code", methods=["GET", "POST"])
def sms_code():
    # 获取前段传递过来的json数据
    mobile = request.json.get("mobile")
    # 参数表示图片验证码的内容
    image_code = request.json.get("image_code")
    # 表示图片验证码的id
    image_code_id = request.json.get("image_code_id")
    # 校验前端传递过来的数据是否有值
    if not all([mobile, image_code, image_code_id]):
        return jsonify(errno=RET.PARAMERR, errmsg="请输入参数")
    # 校验用户传递过来的手机号是否正确
    if not re.match("1[35678]\d{9}", mobile):
        return jsonify(erron=RET.PARAMERR, errmsg="请输入正确的手机号")
    # 获取到redis里面的图片验证码
    real_image_code = redis_store.get("image_code_" + image_code_id)
    # 判断redis是否过期
    if not real_image_code:
        return jsonify(erron=RET.PARAMERR, errmsg="图片验证码已过期")

    # 说明redis没有过期。判断用户输入的验证码，转换成小写再进行判断
    if image_code.lower() != real_image_code.lower():
        return jsonify(erron=RET.PARAMERR, errmsg="请输入正确的图片验证码")

        # 通过随机数生成一个6位的验证码
        # 000001
    random_sms_code = "%06d" % random.randint(0, 999999)
    # 在服务器的redis里面存储短信验证码,用来给用户进行校验操作
    # 第一个参数表示key
    # 第二个参数表示随机数
    # 第三个参数表示过期时间,单位是秒
    redis_store.set("sms_code_" + mobile, random_sms_code, constants.SMS_CODE_REDIS_EXPIRES)

    print("短信验证码的内容是：" + random_sms_code)
    # statucode = CCP().send_template_sms(mobile, [random_sms_code, 5], 1)
    # if statucode != 0:
        # return jsonify(erron=RET.THIRDERR, errmsg="短信发送失败")

    return jsonify(erron=RET.OK, errmsg="发送短信成功")


"""生成图片验证码"""


@passport_blue.route("/image_code")
def image_code():
    print("前端请求的url地址 = " + request.url)
    # 获取到前端传递过来的随机的一个验证码
    code_id = request.args.get("code_id")
    # name:表示图片验证码的名字
    # text:表示图片验证码的内容
    # image:表示是图片
    # 生成图片验证码
    name, text, image = captcha.generate_captcha()
    print("图片验证码的内容 = " + text)
    # 我们需要把图片验证码存到redis的数据库当中
    # image_code_xxxx
    # 第一个参数表示key
    # 第二个参数表示图片验证码的内容
    # 第三个参数表示过期时间
    redis_store.set("image_code_" + code_id, text, constants.IMAGE_CODE_REDIS_EXPIRES)
    # 获取到redis里面的值是byte

    # make_response:表示响应体对象,这个对象的参数表示图片的验证码
    resp = make_response(image)
    # 告诉系统,我们当前需要展示的是图片
    resp.headers['Content-Type'] = 'image/jpg'
    return resp
